

# Dr. D.Y. Patil Institute of Technology, Pimpri, Pune-18.

# **LAB FILE**

**Subject Code/Subject Name** 310248: Lab Practice-I (LP-1 Lab)

# Course/Branch TE Computer Engineering 2019 Pattern

**Division: TE-ALL** 

# **Executed By**

Asst. Prof. Adsure Sharad Asst. Prof. Patil Sunita Asst. Prof. Reddy Kamal Asst. Prof. Shahkar Minal Asst. Prof. Shrikant B. V.

# H.O.D. Computer Dept.

Dr. Chobe S. V.



# Dr.D.Y.Patil Institute of Technology, Pimpri, Pune-411018 Department of Computer Engineering

# Lab Plan

Class: TE Computer Semester and Year: Sem-V (2022-2023)

Subject: LP-I Examination Scheme:
Teaching Scheme : Term Work: 25 Marks
Practical: 4 hrs/week Practical: 25 Marks

| Practical<br>No. | Assignments                                                                                                                                                                                                                                                                                        | Teaching Media /<br>Methodology                        | References                                                                                                                                            |
|------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------|
|                  | Group A                                                                                                                                                                                                                                                                                            |                                                        |                                                                                                                                                       |
| 1                | Design suitable Data structures and implement Pass-I of a<br>two-pass assembler for pseudo-machine. Implementation<br>should consist of a few instructions from each category and<br>few assembler directives                                                                                      | Visualization<br>(Use of video and PPT)                | https://www.youtube.com/watc<br>h?v=0j1XDpAtcWU<br>https://www.youtube.com/watc<br>h?v=dMCuHpxMV8U<br>https://www.youtube.com/watc<br>h?v=a8hXmcdsdTQ |
| 2                | Design suitable Data structures and implement Pass-II of a two-pass assembler for pseudo-machine. Implementation should consist of a few instructions from each category and few assembler directives. The output of Pass-I (intermediate code file and symbol table) should be input for Pass-II. | Visualization (Use of video and PPT)                   | https://www.youtube.com/watc<br>h?v=EcH6DevLln0<br>https://www.youtube.com/watc<br>h?v=27G06PxYKv0<br>https://www.youtube.com/watc<br>h?v=AATIeMs3T7s |
| 3                | Design suitable data structures and implement Pass-I of a two-pass macro-processor. The output of Pass-I (MNT, MDT and intermediate code file without any macro definitions) should be input for Pass-II.                                                                                          | Visualization<br>(Use of video and PPT)                | https://www.youtube.com/watc<br>h?v=PCbDjGBybLo                                                                                                       |
| 4                | Design suitable data structures and implement Pass-I and Pass-II of a two-pass macro-processor. The output of Pass-I (MNT, MDT and intermediate code file without any macro definitions) should be input for Pass-II.                                                                              | Visualization<br>(Use of video and PPT)                | https://www.youtube.com/watc<br>h?v=5uuPjYXSf88<br>https://www.youtube.com/watc<br>h?v=xZp7cV6ty0I                                                    |
| 5                | Write a program to create a Dynamic Link Library for any mathematical operation and write an application program to test it. (Java Native Interface / Use VB or VC++)                                                                                                                              | Visualization<br>(Use of video and PPT)                | https://www.youtube.com/watc<br>h?v=eW5he5uFBNM<br>https://www.youtube.com/watc<br>h?v=PUVPuGQ9P5Y                                                    |
|                  | Group B                                                                                                                                                                                                                                                                                            |                                                        |                                                                                                                                                       |
| 6                | Write a program to solve Classical Problems of Synchronization using Mutex and Semaphore                                                                                                                                                                                                           | Inquiry-Based<br>Instruction                           | R2                                                                                                                                                    |
| 7                | Write a program to simulate CPU Scheduling Algorithms: FCFS, SJF (Preemptive), Priority (Non-Preemptive) and Round Robin (Preemptive).                                                                                                                                                             | Inquiry-Based<br>Instruction,<br>Experimental Learning | https://www.youtube.com/watc<br>h?v=zoRYtf0856k<br>https://www.youtube.com/watc<br>h?v=R8LYxTT9wmY&t=461s<br>https://www.youtube.com/watc             |

|    |                                                                                                                                                                                                                                                                                                                                                                  |                                                        | h?v=1ikcIKDGEz4                                                                                    |
|----|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------|----------------------------------------------------------------------------------------------------|
| 8  | Write a program to simulate Memory placement strategies – best fit, first fit, next fit and worst fit.                                                                                                                                                                                                                                                           | Inquiry-Based Instruction, Experimental Learning       | https://www.youtube.com/wate<br>h?v=8M0o2FH3RUo                                                    |
| 9  | Write a program to simulate Page replacement algorithm.                                                                                                                                                                                                                                                                                                          | Inquiry-Based Instruction, Experimental Learning       | https://www.youtube.com/wate<br>h?v=Ub4VVDGLJx0<br>https://www.youtube.com/wate<br>h?v=UTXkbcJUY74 |
| 10 | Write a program to implement Deadlock Avoidance<br>Algorithm                                                                                                                                                                                                                                                                                                     | Inquiry-Based<br>Instruction,<br>Experimental Learning | https://www.youtube.com/wat<br>h?v=w0LwGqffUkg                                                     |
|    | Group C                                                                                                                                                                                                                                                                                                                                                          |                                                        |                                                                                                    |
| 11 | Implementation of Inter-process communication using socket programming: implementing multithreaded echo server.                                                                                                                                                                                                                                                  | Visualization (Use of video and PPT)                   | R5, R6                                                                                             |
| 13 | Simulation of election algorithms (Ring and Bully).                                                                                                                                                                                                                                                                                                              | Visualization (Use of video and PPT)                   | R5, R6                                                                                             |
|    | Group D                                                                                                                                                                                                                                                                                                                                                          |                                                        |                                                                                                    |
| 14 | Create Project Plan  Specify project name and start (or finish) date.  Identify and define project tasks.  Define duration for each project task.  Define milestones in the plan  Define dependency between tasks  Define project calendar.  Define project resources and specify resource type Assign resources against each task and baseline the project plan | Inquiry-Based Instruction, Experimental Learning       | R3, R4                                                                                             |
| 15 | Execute and Monitor Project Plan  Update % Complete with current task status.  Review the status of each task.  Compare Planned vs Actual Status  Review the status of Critical Path Review resources assignation status                                                                                                                                         | Inquiry-Based Instruction, Experimental Learning       | R3, R4                                                                                             |

### **Reference Books:**

- 1. Dhamdhere D., "Systems Programming and Operating Systems", McGraw Hill, ISBN 0 07 463579
- 2. Silberschatz, Galvin, Gagne, "Operating System Principles", 9th Edition, Wiley, ISBN 978-1-118-06333-0
- 3. Bob Hughes, Mike Cotterell and Rajib Mall, "Software Project Management", Sixth Edition, Tata McGraw Hill, New Delhi, 2017
- 4. Robert K. Wysocki, "Effective Software Project Management", Wiley Publication, 2011
- 5. Maarten van Steen, Andrew S. Tanenbaum, "Distributed System", Third edition, version 3
- 6. George Coulouris, Jean Dollimore, Tim Kindberg, "Distributed Systems Concepts and Design", Fifth edition

# WebLinks:

- 1. <a href="https://www.youtube.com/watch?v=VG9VopzV\_T0">https://www.youtube.com/watch?v=VG9VopzV\_T0</a>
- 2. <a href="https://www.youtube.com/watch?v=0j1XDpAtcWU">https://www.youtube.com/watch?v=0j1XDpAtcWU</a>
- 3. <a href="https://www.youtube.com/watch?v=zoRYtf0856k">https://www.youtube.com/watch?v=zoRYtf0856k</a>
- 4. <a href="https://www.youtube.com/watch?v=Ub4VVDGLJx0">https://www.youtube.com/watch?v=Ub4VVDGLJx0</a>
- 5. https://www.youtube.com/watch?v=8M0o2FH3RUo
- 6. <a href="https://www.youtube.com/watch?v=8dc6zz6fkDA">https://www.youtube.com/watch?v=8dc6zz6fkDA</a>

# **ASSIGNMENT NO: 1 – Group A**

# **Title of Assignment:**

Design suitable Data structures and implement Pass-I of a two-pass assembler for pseudo-machine.

## **Problem Statement:**

Implement one pass-I of TWO Pass assembler with hypothetical Instruction set using Java language. Instruction set should include all types of assembly language statements such as Imperative, Declarative and Assembler Directive. While designing stress should be given on

- a) How efficiently Mnemonic opcode could be implemented so as to enable faster retrieval on op-code.
- b) Implementation of symbol table for faster retrieval.

**Objectives:** learn how pass-1 and pass-2 data structure works in assembly programs input.

# Theory: Assembler:

- 1. An assembler is a program that accepts as input an assembly language program and produces as output its machine language equivalent i.e. It produces bit configuration of each of the mnemonic in the assembly language as shown. This machine language information from assembler is given to loader for further processing.
- 2 Note that in std. Cases Assembler not only produce this bit configuration; but also produces information useful for loader like- The externally defined symbols etc. are noted and these symbols are passed on to the loader for further resolution of their addresses.
- 3. The main reason for existence of assembler was to shift the burdens of calculating specific addresses from programmer to computer.

# Tasks performed by the pass2 assembler

## **Pass 1:**

- 1. Separate symbol, Mnemonic and Operand fields
- 2. Build the symbol table
- 3. Perform IC processing
- 4. Construct intermediate code

# Pass 2:

- 1. Synthesis of target program
- 2. Evaluate fields and generate code
- 3. Process pseudo opcodes

## Pass 1 of assembler:

- 1. Pass1 uses following data structures:
  - a. Symbol table (st).
  - b. Literal table(lt)
  - c. Machine opcode table(mot)
  - d. Location counter
  - e. Copy of source program
- 2. In some designs, assembler uses following table (i)Base Table BT:

- (a) To store the contents of base register
- (b) This table is used when we encounter the USING and BALR directives (ii)Pool table(POOL TAB):
  - a) Awareness of different literal pools is maintained using auxiliary table POOL TAB
  - b) This table contains literal number of the starting literal of each pool
  - c) At any stage the current literal pool is last pool in the literal table
  - d) In encountering an LTORG statement literals in the current pool are allocated addresses starting with current value in LC and LC is appropriately incremented
- 3. Now the principle activities of pass 1 are:
  - (i) Find length of machine instruction from MOT
  - (ii) Keep track of location counter
  - (iii) Remember values of symbols until pass II
  - (iv) Process some pseudo-op-code
  - (v)Remember literals

# **Algorithm for Pass I:**

```
a. oc_cntr := 0; (default value)
       pooltab_ptr :=1; POOLTAB[1]:=1;
       littab ptr:=1;
b. While next statement is not an END statement
       a) If label is present then
       this label:= symbol in label field;
       Enter(this_label, loc_cntr) in SYMTAB.
       b) If an LTORG statement then
              i Process literals
               LITTAB[POOLTAB[pooltab ptr]...LITTAB[lit tab ptr-1] to
               allocate memory and put the address in the address field.
               Update location counter accordingly.
              ii pooltab_ptr := pooltab_ptr +1;
              iii POOLTAB[pooltab_ptr]:=littab_ptr;
       c) If START or ORIGIN statement then
       loc_cntr := value specified in the operand field;
       d) If an EQU statement then
              i. this addr := value of <address spec>;
               ii. Correct the symbtab entry for this_label to
               (this label, this addr).
       e) If a declaration statement then
              i. code:= code of the declaration statement;
              ii. size := size of memory are required by DC/DS
              iii. loc_cntr := loc_cntr + size;
```

```
iv. Generate IC '(DL, code)...'
             f) If an imperative statement then
                    i. code:= machine opcode from OPTAB;
                    ii. loc_cntr := loc_cntr + instruction length from OPTAB;
                    iii. If operand is a literal then
                    this_literal := literal in operand field;
                    LITTAB[littab_ptr]:= this_literal;
                    littab_ptr= littab_ptr +1;
                    }
                    else (i.e. operand is a symbol)
                    this_entry := SYMTAB entry number of operand
                    Generate IC '(IS,code)(S,this entry)';
      c.
             a) Perform step 2(b).
             b) Generate IC'(AD, 02)'.
             c) Go to Pass II.
SAMPLE PROGRAM Input
START 200
READ A
READ B
MOVER AREG, ='5'
MOVER AREG, A
ADD AREG, B
SUB AREG, ='6'
MOVEM AREG, C
PRINT C
LTORG
MOVER AREG, ='15'
MOVER AREG, A
ADD AREG, B
SUB AREG, ='16'
DIV AREG, ='26'
MOVEM AREG, C
ADS 1
BDS 1
CDS1
STOP
END Symbol Table
Symb Addr Decl Used Val Len
A 216 1 1 0 1
B 217 1 1 0 1
C 218 1 1 0 1
```

Total Errors: 0 Total Warnings: 0 Literal Table \_\_\_\_\_ Lit# Lit Addr 00 = '5' 20801 = 6' 209 02 = '15' 220 03 = '16' 221 04 = '26' 222 Pool Table -----Pool# Pool Base \_\_\_\_\_ 000012

# INTERMEDIATE CODE

Intermediate Code

(AD, 00) (C, 200)

(IS, 09)(S, 00)

(IS, 09)(S, 01)

(IS, 04)(0)(L, 00)

(IS, 04) (0) (S, 00)

(IS, 01)(0)(S, 01)

(IS, 02) (0) (L, 01)

(IS, 05) (0) (S, 02)

(IS, 10) (S, 02)

(AD, 04)

(IS, 04) (0) (L, 02)

(IS, 04)(0)(S, 00)

(IS, 01)(0)(S, 01)

(IS, 02)(0)(L, 03)

(IS, 08) (0) (L, 04)

(IS, 05)(0)(S, 02)

(DL, 01) (C, 01)

(DL, 01) (C, 01)

(DL, 01) (C, 01)

(IS, 00)

(AD, 01)

# **Conclusion:**

# **ASSIGNMENT NO: 2 – Group A**

# **Title of Assignment:**

Design suitable Data structures and implement Pass-II of a two-pass assembler for pseudo-machine.

## **Problem Statement:**

Implement pass-II of TWO Pass assembler with hypothetical Instruction set using Java language. Instruction set should include all types of assembly language statements such as Imperative, Declarative and Assembler Directive. While designing stress should be given on

- a) How efficiently Mnemonic opcode table could be implemented so as to enable faster retrieval on opcode.
- b) Implementation of symbol table, pool tables for faster retrieval.

**Objectives:** learn how pass-1 and pass-2 data structure works in assembly programs input.

# Theory: Assembler:

# **ALGORITHM:**

- 1. Open and read the first line from the intermediate file.
- 2. If the first line contains the opcode "START", then write the label, opcode and operand field values of the corresponding statement directly to the final output file.
- 3. Do the following steps, until an "END" statement is reached.
  - 3.1 Start writing the location counter, opcode and operand fields of the corresponding statement to the output file, along with the object code.
  - 3.2 If there is no symbol/label in the operand field, then the operand address is assigned as zero and it is assembled with the object code of the instruction.
  - 3.3 If the opcode is BYTE, WORD, RESB etc convert the constants to the object code.
- 4. Close the files and exit.

**Forward Referencing**: - In forward referencing, variable or label is referenced before it is declared. Different problems can be solved using **One Pass** or **Two Pass** forward referencing. In One Pass forward referencing source program is translated instruction by instruction. Assembler leave address space for label when it is referenced and when assembler found the declaration of label, it uses *back patching*.

Two Pass forward referencing consist of two passes.

During first pass *symbol table*, *op-code table* and *label table* are maintained.

- In op-code table instruction size and address is stored.
- Label and label's address is stored in label table. When label is encountered, its name is stored in label table when label declaration is found then its location is also stored in label table.

During 2nd Pass, translation from source language to machine language takes place. Instruction addresses and label addresses are used from symbol table instead of their names.

Compiler does not know where program will be executed in the memory so compiler generated logical addresses instead of absolute address. Loader also uses the *Relocation Constant* to solve the problem of relocation. **External Referencing** problem is resolved by the linker during compilation. Linker connects the object program to the code for standard library functions.

# The assembler implements the back patching technique as follows:

- It builds a table of incomplete instructions (TII) to record information about instructions whose operand fields left blank.
- Each entry in this table contains a pair of the form (instruction address, symbol) to indicate that the address of symbol should put in the operand field of the instruction with the address instruction address.
- By the time the END statement processed, the symbol table would contain the addresses of all symbols defined in the source program and TII would contain information describing all forward references.
- The assembler can now process each entry in TII to complete the concerned instruction.
- Alternatively, entries in TII can process on the fly during normal processing.
- In this approach, all forward references to a symbol i would be processed when the statement that defines symbol encountered. also ...
- The instruction corresponding to the statement
  - MOVER BREG, ONE contains a forward reference to ONE.
- Hence the assembler leaves the second operand field blank in the instruction that assembled to reside in location 101 of memory and makes an entry (101, ONE) in the table of incomplete instructions (TII).
- While processing the statement ONE DC '1'address of ONE, which 115, entered in the symbol table.
- After the END statement processed, the entry (101, ONE) would be processed by obtaining the address of ONE from the symbol table and inserting it in the second operand field of the instruction with assembled address 101.

# **Pass II Algorithm**

It has been assumed that the target code is to be assembled in the named code\_area.

1. code\_area\_address := address of code\_area;

```
Pooltab_ptr :=1;
Loc cntr:=0;
```

- 2. While next statement is not an END statement
  - (a) Clear machine\_code\_buffer;
  - (b) If an LTORG statement
    - (i) Process literals in LITTAB[POOLTAB[pooltab ptr]]...
    - LITTAB[POOLTAB[pooltab\_ptr+1]]-1 similar to processing of constants in a DC statement i.e. assemble the literals in machine\_code\_buffer.
    - (ii) size := size of memory area required for literals;
    - (iii) pooltab\_ptr:= pooltab\_ptr +1;
  - (c) If a START or ORIGIN statement then
    - (i) loc\_cntr := value specified in operand field;
    - (ii) size:=0;
  - (d) If a declaration statement

(i) If a DC statement then

Assemble the constant in machine\_code\_buffer.

- (ii) size: = size of memory area required by DC/DS;
- (e) If an imperative statement
  - (i) Get operand address from SYMTAB or LITTAB.
  - (ii) Assemble instruction in machine\_code\_buffer.
  - (iii) size: = size of instruction;
- (f) if size not equal to 0 then
  - (i) Move contents of Machine\_code\_buffer to the address

code\_area\_address + loc\_cntr;

- (ii) loc\_cntr := loc\_cntr + size;
- 3. (Processing of END statement)
  - (a) Perform steps 2(b) and 2(f).
  - (b) Write code\_area into output file.

# **Example:**

# **SAMPLE PROGRAM Input**

**START 200** 

READ A

**READ B** 

MOVER AREG, ='5'

MOVER AREG, A

ADD AREG, B

SUB AREG, ='6'

MOVEM AREG, C

PRINT C

**LTORG** 

MOVER AREG, ='15'

MOVER AREG, A

ADD AREG, B

**SUB AREG**, ='16'

DIV AREG, ='26'

MOVEM AREG, C

A DS 1

BDS 1

CDS1

**STOP** 

**END Symbol Table** 

| Symb | Addr Decl | Used | Val | Len |
|------|-----------|------|-----|-----|
|      |           |      |     |     |
| A    | 216       | 1    | 10  | 1   |
| В    | 217       | 1    | 10  | 1   |
| C    | 218       | 1    | 10  | 1   |
|      |           |      |     |     |

Total Errors: 0

Total Warnings: 0

# Literal Table

-----

# Lit# Lit Addr

-----

00 = '5' 208

01 = 6' 209

02 = '15' 220

03 = '16' 221

04 = '26' 222

-----

## Pool Table

\_\_\_\_\_

# Pool# Pool Base

-----

000

01 2

-----

# INTERMEDIATE CODE

Intermediate Code

(AD, 00) (C, 200)

(IS, 09)(S, 00)

(IS, 09)(S, 01)

(IS, 04)(0)(L, 00)

(IS, 04) (0) (S, 00)

(IS, 01)(0)(S, 01)

(IS, 02) (0) (L, 01)

(IS, 05) (0) (S, 02)

(IS, 10) (S, 02)

(15, 10) (5, 02)

(AD, 04)

(IS, 04)(0)(L, 02)

(IS, 04)(0)(S, 00)

(IS, 01) (0) (S, 01)

(IS, 02) (0) (L, 03)

(IS, 08)(0)(L, 04)

(IS, 05) (0) (S, 02)

(DL, 01) (C, 01)

(DL, 01) (C, 01)

(DL, 01) (C, 01)

(IS, 00)

(AD, 01)

# **PASS II OUTPUT**

Target Code

200) + 090216

201) + 090217

202) + 04 0 208 203) + 04 0 216

204) + 01 0 217

205) + 02 0 209

206) + 050218

207) + 100218208) + 0000005209) + 0000006210) + 040220211) + 040216 $212) + 01 \ 0 \ 217$ 213) + 020221214) + 080222215) + 050218216) 217) 218) 219) + 0000000220) + 0000015221) + 000016222) + 0000026

# **Conclusion:**

# **ASSIGNMENT NO: 3 – Group A**

# **Title of Assignment:**

Design suitable data structures and implement pass-I of a two-pass macro-processor.

**Problem Statement:** Write a program in C for a pass-II of two pass macro processor for Implementation of Macro Processor. Following cases to be considered

- a) Macro without any parameters
- b) Macro with Positional Parameters
- c) Macro with Key word parameters
- d) Macro with positional and keyword parameters.

(Conditional expansion, nested macro implementation not expected)

**Objectives:** learn how pass-1 and pass-2 Macro processor works.

**Theory: Macro Processor:** A macro processor is a program that copies a stream of text from one place to another, making a systematic set of replacements as it does so. Macro processors are often embedded in other programs, such as assemblers and compilers. Sometimes they are standalone programs that can be used to process any kind of text. "A macro processor is a program that reads a file (or files) and scans them for certain keywords. When a keyword is found, it is replaced by some text. The keyword/text combination is called a macro."

Two new assembler directives are used in macro definition are MACRO and MEND.

MACRO: identify the beginning of a macro definition

MEND: identify the end of a macro definition

Prototype for the macro Each parameter begins with '&'

```
name MACRO parameters
: body
: MEND
```

Body: the statements that will be generated as the expansion of the macro.

# Macro facility:

An assembly language macro facility is to extend the set of operations provided in an assembly language. In order that programmers can repeat identical parts of their program macro facility can be used. This permits the programmer to define an abbreviation for a part of program & use this abbreviation in the program. This abbreviation is treated as macro definition & saved by the macro processor. For all occurrences the abbreviation i.e. macro call, macro processor, substitutes the definition.

Data Structure required for macro definition processing:

- Macro Name Table (MNT):- Fields Name of macro, #PP(Number of positional parameters),#KP(Number of keyword parameters), MDTP(Macro Definition Table Pointer), KPDTP (Keywords Parameters Default Table Position).
- Parameter Name Table (PNTAB) :- Feilds parameter name
- Keywords Parameters Default Table (KPDTAB) :- Feilds-parameter name, default value.
- Macro Definition Table (MDT): Model statements are stored in inetrmediate code form as: opcode and operands.





\_\_\_\_\_

- 0 INCR &X, &Y,&REG=AREG
- 1 MOVER #2,#0
- 2 ADD #2,#1
- 3 MOVEM #2,#0
- 4 MEND
- 5 DECR &A,&B,&REG=BREG
- 6 MOVER #2,#0
- 7 SUB #2,#1
- 8 MOVEM #2,#0
- 9 MEND

-----

### Contents of test.asm

-----

**MACRO** 

INCR &X, &Y,&REG=AREG

MOVER & REG, & X

ADD &REG,&Y

MOVEM & REG, & X

**MEND** 

**MACRO** 

DECR &A,&B,&REG=BREG

MOVER & REG, & A

SUB &REG,&B

MOVEM &REG,&A

**MEND** 

**START 100** 

READ N1

READ N2

INCR N1,N2,REG=CREG

DECR N1,N2

**STOP** 

N1 DS 1

N2 DS 1

**END** 

# Contents of test.ini

-----

START 100

READ N1

**READ N2** 

- +MOVER CREG,N1
- +ADD CREG,N2
- +MOVEM CREG,N1
- +MOVER BREG,N1
- +SUB BREG,N2
- +MOVEM BREG,N1

**STOP** 

N1 DS 1

N2 DS 1

**END** 

**Conclusion:** Thus we have implemented pass-I of two pass Macro processor.

# **ASSIGNMENT NO: 4 – Group A**

# **Title of Assignment:**

Write a Java program for pass-II of a two-pass macro-processor.

# **Problem Statement:**

Implement pass-II of TWO Pass assembler with hypothetical Instruction set using Java language. Instruction set should include all types of assembly language statements such as Imperative, Declarative and Assembler Directive. While designing stress should be given on

- a) How efficiently Mnemonic opcode table could be implemented so as to enable faster retrieval on opcode.
- b) Implementation of symbol table, pool tables for faster retrieval.

**Objectives:** learn how pass-1 and pass-2 data structure works in assembly programs input.

**Theory: Macros** are compiled programs that you can invoke (or **call**) in a submitted SAS program or from a SAS command prompt. Like macro variables, you generally use macros to generate text. However, macros provide additional capabilities:

- Macros can contain programming statements that enable you to control how and when text is generated.
- Macros can accept parameters. This allows you to write generic macros that can serve a number of uses.

To compile a macro, you must submit a macro definition. The general form of a macro definition is

%MACRO macro-name; <macro\_text> %MEND <macro\_name>;

where **macro\_name** is a unique SAS name that identifies the macro and **macro\_text** is any combination of macro statements, macro calls, text expressions, or constant text.

When you submit a macro definition, the macro processor compiles the definition and produces a member in the session catalog. The member consists of compiled macro program statements and text. The distinction between compiled items and noncompiled (text) items is important for macro execution. Examples of text items include:

- macro variable references
- nested macro calls
- macro functions, except %STR and %NRSTR
- arithmetic and logical macro expressions
- text to be written by %PUT statements
- field definitions in %WINDOW statements
- model text for SAS statements and SAS Display Manager System commands.

When you want to call the macro, you use the form

**Macro expansion:-** A macro name is an abbreviation, which stands for some related lines of code. Macros are useful for the following purposes:

- · To simplify and reduce the amount of repetitive coding
- · To reduce errors caused by repetitive coding
- · To make an assembly program more readable.

A macro consists of name, set of formal parameters and body of code. The use of macro name with set of actual parameters is replaced by some code generated by its body. This is called macro expansion.

Macros allow a programmer to define pseudo operations, typically operations that are generally desirable, are not implemented as part of the processor instruction, and can be implemented as a sequence of instructions. Each use of a macro generates new program instructions, the macro has the effect of automating writing of the program.

Macros can be defined used in many programming languages, like C, C++ etc. Example macro in C programming. Macros are commonly used in C to define small snippets of code. If the macro has parameters, they are substituted into the macro body during expansion; thus, a C macro can mimic a C function. The usual reason for doing this is to avoid the overhead of a function call in simple cases, where the code is lightweight enough that function call overhead has a significant impact on performance.

For instance, #define max (a, b) a>b? A: b

Defines the macro max, taking two arguments a and b. This macro may be called like any C function, using identical syntax. Therefore, after preprocessing

```
z = max(x, y); Becomes z = x>y? X:y;
```

While this use of macros is very important for C, for instance to define type-safe generic data-types or debugging tools, it is also slow, rather inefficient, and may lead to a number of pitfalls.

C macros are capable of mimicking functions, creating new syntax within some limitations, as well as expanding into arbitrary text (although the C compiler will require that text to be valid C source code, or else comments), but they have some limitations as a programming construct. Macros which mimic functions, for instance, can be called like real functions, but a macro cannot be passed to another function using a function pointer, since the macro itself has no address.

In programming languages, such as C or assembly language, a name that defines a set of commands that are substituted for the macro name wherever the name appears in a program (a process called macro expansion) when the program is compiled or assembled. Macros are similar to functions in that they can take arguments and in that they are calls to lengthier sets of instructions. Unlike functions, macros are replaced by the actual commands they represent when the program is prepared for execution. function instructions are copied into a program only once.

**Macro Expansion:-** A macro call leads to macro expansion. During macro expansion, the macro statement is replaced by sequence of assembly statements. In the above program a macro call is shown in the middle of the figure. i.e. INITZ. Which is called during program execution? Every macro begins with MACRO keyword at the beginning and ends with the ENDM (end macro).whenever a macro is called the entire is code is substituted in the program where it is called. So the resultant of the macro code is shown on the right most side of the figure. Macro calling in high level programming languages

The above program was written using C programming statements. Defines the macro max, taking two arguments a and b. This macro may be called like any C function, using identical syntax. Therefore, after preprocessing Becomes z = x>y? x: y;

After macro expansion, the whole code would appear like this.

```
#define max(a,b) a>b?a:b

main()

{

int x , y;

x=4; y=6;

z = x>y?x:y;

}
```



Conclusion:- Thus we have implemented pass-II of two pass Macro processor

# **ASSIGNMENT NO: 5 – Group A**

**Title of Assignment:** Write a program to create Dynamic Link Library for any mathematical operation and write an application program to test it. (Java Native Interface / Use VB or VC++).

Problem Statement: Write a program to create Dynamic Link Library for Arithmetic Operation in VB.net

# **Objectives:**

- To understand Dynamic Link Libraries Concepts
- To implement dynamic link library concepts
- To study about Visual Basic

# **Theory:**

**Dynamic Link Library:** A dynamic link library (DLL) is a collection of small programs that can be loaded when needed by larger programs and used at the same time. The small program lets the larger program communicate with a specific device, such as a printer or scanner. It is often packaged as a DLL program, which is usually referred to as a DLL file. DLL files that support specific device operation are known as device drivers.

A DLL file is often given a ".dll" file name suffix. DLL files are dynamically linked with the program that uses them during program execution rather than being compiled into the main program.

The advantage of DLL files is space is saved in random access memory (RAM) because the files don't get loaded into RAM together with the main program. When a DLL file is needed, it is loaded and run. For example, as long as a user is editing a document in Microsoft Word, the printer DLL file does not need to be loaded into RAM. If the user decides to print the document, the Word application causes the printer DLL file to be loaded and run.

A program is separated into modules when using a DLL. With modularized components, a program can be sold by module, have faster load times and be updated without altering other parts of the program. DLLs help operating systems and programs run faster, use memory efficiently and take up less disk space

# **Feature of DLL**

- DLLs are essentially the same as EXEs, the choice of which to produce as part of the linking process is for clarity, since it is possible to export functions and data from either.
- It is not possible to directly execute a DLL, since it requires an EXE for the operating system to load it through an entry point, hence the existence of utilities like RUNDLL.EXE or RUNDLL32.EXE which

provide the entry point and minimal framework for DLLs that contain enough functionality to execute without much support.

- DLLs provide a mechanism for shared code and data, allowing a developer of shared code/data to upgrade functionality without requiring applications to be re-linked or re-compiled. From the application development point of view Windows and OS/2 can be thought of as a collection of DLLs that are upgraded, allowing applications for one version of the OS to work in a later one, provided that the OS vendor has ensured that the interfaces and functionality are compatible.
- DLLs execute in the memory space of the calling process and with the same access permissions which means there is little overhead in their use but also that there is no protection for the calling EXE if the DLL has any sort of bug.

### **Executable file links to DLL:**

An executable file links to (or loads) a DLL in one of two ways:

**Implicit linking:** - Implicit linking is sometimes referred to as static load or load-time dynamic linking. Explicit linking is sometimes referred to as dynamic load or run-time dynamic linking. With implicit linking, the executable using the DLL links to an import library (.lib file) provided by the maker of the DLL. The operating system loads the DLL when the executable using it is loaded. The client executable calls the DLL's exported functions just as if the functions were contained within the executable

**Explicit linking:-**With explicit linking, the executable using the DLL must make function calls to explicitly load and unload the DLL and to access the DLL's exported functions. The client executable must call the exported functions through a function pointer. An executable can use the same DLL with either linking method. Furthermore, these mechanisms are not mutually exclusive, as one executable can implicitly link to a DLL and another can attach to it explicitly.

# **Calling DLL function from Visual Basic Application:**

For Visual Basic applications (or applications in other languages such as Pascal or Fortran) to call functions in a C/C++ DLL, the functions must be exported using the correct calling convention without any name decoration done by the compiler.

\_\_stdcall creates the correct calling convention for the function (the called function cleans up the stack and parameters are passed from right to left) but decorates the function name differently. So, when \_\_declspec(dllexport) is used on an exported function in a DLL, the decorated name is exported.

The \_\_stdcall name decoration prefixes the symbol name with an underscore (\_) and appends the symbol with an at sign (@) character followed by the number of bytes in the argument list (the required stack space). As a result, the function when declared as:

int \_\_stdcall func (int a, double b)

is decorated as:

\_func@12

The C calling convention (\_\_cdecl) decorates the name as \_func.

To get the decorated name, use /MAP. Use of \_\_declspec(dllexport) does the following:

- If the function is exported with the C calling convention (**\_cdecl**), it strips the leading underscore (**\_)** when the name is exported.
- If the function being exported does not use the C calling convention (for example, \_\_stdcall), it exports the decorated name.

Because there is no way to override where the stack cleanup occurs, you must use \_\_stdcall. To undecorate names with \_\_stdcall, you must specify them by using aliases in the EXPORTS section of the .def file. This is shown as follows for the following function declaration:

int \_\_stdcall MyFunc (int a, double b);

void \_\_stdcall InitCode (void);

In the .DEF file:

**EXPORTS** 

MYFUNC=\_MyFunc@12

INITCODE=\_InitCode@0

For DLLs to be called by programs written in Visual Basic, the alias technique shown in this topic is needed in the .def file. If the alias is done in the Visual Basic program, use of aliasing in the .def file is not necessary. It can be done in the Visual Basic program by adding an alias clause to the Declare statement.

**Conclusion:-** Thus, I have studied visual programming and implemented dynamic link library application for arithmetic operation

## **Ouestions:**

- 1. What Is Dll And What Are Their Usages And Advantages?
- **2.** What Are The Sections in a DLL Executable/binary?
- **3.** Where Should We Store DLL?
- **4.** Who Loads and Links the DLL?
- **5.** How Many Types Of Linking Are There?
- **6.** What Is Implicit And Explicit Linking In Dynamic Loading?

# **ASSIGNMENT NO: 06 – Group B**

**Assignment Title:** Implement a program to solve Classical Problems of Synchronization using Mutex and Semaphore

**Problem Statement:** Write a program to solve Classical Problems of Synchronization using Mutex and Semaphore

# **Objectives:**

- 1. To understand concept of semaphore, critical section.
- 2. Learn implementation of Semaphore.

# **Theory:**

- There are a number of processes that only read the data area (readers) and a number that only write to the data area (writers).
- The conditions that must be satisfied are
  - o Any number of readers may read simultaneously read the file.
  - Only one write at a time may write to the file.
  - o If a writer is writing to the file, no reader may read it.
- The data area could be a file, a block of main memory, or even a bank of processor registers.
- There is a data area shared among a number of processor registers

**Semaphore:** Semaphores are system variables used for synchronization of process.

# • Binary Semaphore –

This is also known as mutex lock. It can have only two values -0 and 1. Its value is initialized to 1. It is used to implement the solution of critical section problems with multiple processes.

# • Counting Semaphore –

Its value can range over an unrestricted domain. It is used to control access to a resource that has multiple instances.

```
P(Semaphore s){
while(S == 0); /* wait until s=0 */
s=s-1;
}

Note that there is
Semicolon after while.
The code gets stuck
Here while s is 0.
```

```
// Some code
P(s);
// critical section
V(s);
// remainder section
```

- P operation is also called wait, sleep, or down operation, and V operation is also called signal, wake-up, or up operation.
- Both operations are atomic and semaphore(s) is always initialized to one. Here atomic means that variable on which read, modify and update happens at the same time/moment with no pre-emption i.e. in-between read, modify and update no other operation is performed that may change the variable.

• A critical section is surrounded by both operations to implement process synchronization. See the below image. The critical section of Process P is in between P and V operation.

# **Algorithm for Reader Writer:**

- 1. import java.util.concurrent.Semaphore;
- 2. Create a class RW
- 3. Declare semaphores mutex and wrt
- 4. Declare integer variable readcount = 0
- 5. Create a nested class Reader implements Runnable
  - a. Override run method (Reader Logic)
    - i. wait(mutex);
    - **ii.** readcount := readcount +1;
    - **iii.** if readcount = 1 then
    - iv. wait(wrt);
    - **v.** signal(mutex);
    - vi. ...
    - vii. reading is performed
    - viii. ...
      - ix. wait(mutex);
      - **x.** readcount := readcount -1;
      - **xi.** if readcount = 0 then signal(wrt);
    - xii. signal(mutex):
- 6. Create a nested class Writer implements Runnable
  - a. Override run method (Writer Logic)
    - i. wait(wrt);
    - ii. ...
    - iii. writing is performed
    - iv. ...
    - v. signal(wrt);
- 7. Create a class main
  - a. Create Threads for Reader and Writer
  - b. Start these thread



# **Limitations:**

- One of the biggest limitations of semaphore is priority inversion.
- Deadlock, suppose a process is trying to wake up another process which is not in a sleep state. Therefore, a deadlock may block indefinitely.
- The operating system has to keep track of all calls to wait and to signal the semaphore.

# **Problem in this implementation of semaphore:**

- The main problem with semaphores is that they require busy waiting,
- If a process is in the critical section, then other processes trying to enter critical section will be waiting until the critical section is not occupied by any process.

  Whenever any process waits then it continuously checks for semaphore value (look at this line while (s==0); in P operation) and waste CPU cycle.
- There is also a chance of "spinlock" as the processes keep on spins while waiting for the lock.

Conclusion: Thus we have learned basic concept of semaphore and mutex.

# **QUESTIONS:**

1. Difference between Binary Semaphore and Mutex

# **ASSIGNMENT NO: 07 – Group B**

**Assignment Title:** implement following process scheduling algorithms: FCFS , SJF (Preemptive), Priority (Non-Preemptive) .

**Problem Statement:** Write a Java program (using OOP features) to implement following scheduling algorithms: FCFS, SJF (Preemptive), Priority (Non-Preemptive).

# **Objectives:**

- 1. To understand concept of scheduling.
- 2. To learn and use scheduling algorithms.

# Theory:

CPU scheduling is the basis of multi programmed operating systems. By switching the CPU among processes, the operating system can make the computer more productive. The objective of multiprogramming is to have some process running at all times, in order to maximize CPU utilization. In a uniprocessor system, only one process may run at a time; any other processes must wait until the CPU is free and can be rescheduled.

The idea of multiprogramming is relatively simple. A process is executed until it must wait, typically for the completion of some I/O request. In a simple computer system, the CPU would then sit idle; all this waiting time is wasted. With multiprogramming, the time is to be used productively. Several processes are kept in memory at one time. When one process has to wait, the operating system takes the CPU away from that process and gives the CPU to another process. This pattern continues.

Scheduling is a fundamental operating system function. Almost all computer resources are scheduled before use. The CPU is one of the primary computer resources. Thus, its scheduling is central to operating system design.

# **CPU-I/O Burst Cycle:**

The success of CPU scheduling depends on the following observed property of processes: Process execution consists of a cycle of CPU execution and I/O wait. Processes alternate between these two states. Process execution begins with a CPU burst i.e. by an I/O burst, and then another CPU burst will end with a system request to terminate execution, rather than with another I/O burst.

# **CPU Scheduler**

Whenever the CPU becomes idle, the operating system must select one of the processes in thread queue to be executed. The selection process is carried out by the short-term scheduler (or CPU scheduler).

The scheduler selects from among the processes in memory that are ready to execute, and allocates the CPU to one of them.

The ready queue is not necessarily a first-in, first-out (FIFO) queue. A ready queue may be implemented as a FIFO queue, a priority queue, a tree, or simply an unordered linked list. All the processes in the ready queue are lined up waiting for a chance to run on the CPU. The records in the queue are generally process control blocks (PCBs) of the processes.

# **Preemptive Scheduling**

CPU scheduling decisions may take place under the following four circumstances:

- 1. When a process switches from the running state to the waiting state (for example, I/O request, or invocation of wait for the termination of one of the child processes)
- 2. When a process switches from the running state to the ready state (for example, when an interrupt occurs)
- 3. When a process switches from the waiting state to the ready state( for example, completion of I/O)
- 4. When a process terminates

In circumstances 1 and 4, there is no choice in terms of scheduling. A new process must be selected for execution. There is a choice in circumstances 2 and 3.

When the scheduling takes place only under circumstances 1 and 4, the scheduling is non-preemptive; otherwise, the scheduling scheme is preemptive. Under non-preemptive scheduling, once the CPU has been allocated to a process, the process keeps the CPU until it releases the CPU either by terminating or by switching to the waiting state. This scheduling method is used by the Microsoft Windows 3.1 and by the Apple Macintosh operating systems. It is the only method that can be used on certain hardware platforms, because it does not require the special hardware needed for preemptive scheduling.

Preemptive scheduling incurs a cost. Consider the case of two processes sharing data. One may be in the midst of updating the data when it is preempted and the second process is run. The second process may try to read the data, which are currently in an inconsistent state. New mechanisms are thus needed to coordinate access to shared date.

Preemption also has an effect on the design of the operating system kernel. During the processing of a system call, the kernel may be busy with an activity on behalf of a process. Such activities may involve changing important kernel data. If the process is preempted in the middle of these changes, and the kernel needs to read or modify the structure, chaos could ensue. Some operating systems, including most version of UNIX, deal with this problem by waiting either for a system call to complete, or for an I/O block to take place, before doing a context switch. This scheme ensures that the kernel structure is simple, since the

kernel will not preempt a process while the kernel data structures are in an inconsistent state. Unfortunately, this kernel execution model is a poor one for supporting real-time computing and multiprocessing.

# First-Come, First-Served Scheduling

The simplest CPU scheduling algorithm is the first-come, first-served (FCFS) scheduling algorithm. With this scheme, the process that requests the CPU first is allocated the CPU first. The implementation of the FCFS policy is easily managed with a FIFO queue. When a process enters the ready queue, its PCB is linked onto the tail of the queue. The running process is then removed from the queue. The code for FCFS scheduling is simple to write and understand.

The average waiting time under the FCFS policy, is often quite long. Consider the following set of processes that arrive at time 0, with the length of the CPU burst time given in milliseconds.

| <u>Process</u> | Burst Time |
|----------------|------------|
| $P_1$          | 24         |
| $P_2$          | 3          |
| $P_3$          | 3          |

If the processes arrive in the order  $P_1$ ,  $P_2$ ,  $P_3$ , and are served in FCFS order, the result is obtained as shown in the Gantt chart:

| $P_1$ | F  | $P_2$ $P_3$ |    |
|-------|----|-------------|----|
| 0     | 24 | 27          | 30 |

The waiting time is 0 milliseconds for process  $P_1$ , 24 milliseconds for process  $P_2$ , and 27 milliseconds for process  $P_3$ . Thus, the average waiting time is (0+24+27)/3=17 milliseconds. If the processes arrive in the order  $P_2$ ,  $P_3$ ,  $P_1$ , the results will be as shown in the following Gantt chart:



The average waiting time is now (0+3+6)/3=3 milliseconds. This reduction is substantial. Thus, the average waiting time under FCFS policy is generally not minimal, and may vary substantially if the process CPU burst times vary greatly.

In addition, consider the performance of FCFS scheduling in a dynamic situation. Assume only one CPU bound process and many I/O bound processes are there. As the processes flow around the system, the following scenario may result. The CPU bound process will get the CPU and hold it. During this time, all the other processes will finish their I/O and move into the ready queue, the I/O devices are idle. Eventually, the CPU bound process finishes its CPU burst and moves to an I/O device. All the I/O bound processes, which have very short CPU bursts, execute quickly and move back to the I/O queues. At this point, the CPU

sits idle. The CPU bound process will then move back to the ready queue and be allocated to the CPU. Again, all the I/O processes end up waiting in the ready queue until the CPU bound process is done. There is a convoy effect, as all the other processes wait for the one big process to get off the CPU. This effect results in lower CPU and device utilization than might be possible if the shorter processes were allowed to go first.

# **Shortest-Job-First Scheduling**

A different approach to CPU scheduling is the shortest-job-first (SJF) scheduling algorithm. This algorithm associates with each process the length of the latter's next CPU burst. When the CPU is available, it is assigned to the process that has the smallest next CPU burst. If two processes have the same length next CPU burst, FCFS scheduling is used to break the tie. The appropriate term would be the *shortest next CPU burst*, because the scheduling is done by examining the length of the next CPU burst of a process, rather than its total length.

Consider the following set of processes, with length of the CPU burst time given in milliseconds as an example:

| <u>Process</u> | <b>Burst Time</b> |
|----------------|-------------------|
| $\mathbf{P}_1$ | 6                 |
| $P_2$          | 8                 |
| $P_3$          | 7                 |
| $\mathbf{P}_4$ | 3                 |

Using SJF scheduling, these processes are scheduled according to the following Gantt chart:



The waiting time is 3 milliseconds for process  $P_1$ , 16 milliseconds for process  $P_2$ , 9 milliseconds for process  $P_3$ , and 0 milliseconds for process  $P_4$ . Thus, the average waiting time is (3+16+9+0)/4=7 milliseconds. If it was FCFS scheduling scheme, then the waiting time would have been 10.25 milliseconds.

The SJF scheduling algorithm is provably optimal, in that it gives the minimum average waiting time for a given set of processes. By moving a short process before a long one, the waiting time of the short process decreases more than it increases the waiting time of the long process. Consequently, the average waiting time decreases.

The real difficulty with the SJF algorithm is to know the length of the next CPU request. For long term (or job) scheduling in a batch system, we can use as the length the process time limit that a user specifies when he submits the job. Thus, users are motivated to estimate the process time limit accurately, since a lower value may mean faster response. SJF scheduling is used frequently in long-term scheduling.

Although the SJF algorithm is optimal, it cannot be implemented at the level of short term CPU scheduling. There is no way to know the length of the next CPU burst. One approach is to try to approximate SJF scheduling. The length of the next CPU burst may not be known, but it can be predicted. It is expected that the next CPU burst will be similar in length to the previous ones. Thus, by computing an approximation of the length of the next CPU burst, the process with the shortest predicted CPU burst can be picked.

The next CPU burst is generally predicted as an exponential average of the measured lengths of previous CPU bursts. Let  $t_n$  be the length of the nth CPU burst, and let  $T_{n+1}$  be the predicted value for the next CPU burst. Then, for  $\alpha$ ,  $0 \le \alpha \le 1$ , define

$$T_{n+1} = \alpha t_n + (1 - \alpha)T_n$$

This formula defines an exponential average. The value of  $t_n$  contains most recent information;  $T_n$  stores the past history in the prediction. The parameter  $\alpha$  controls the relative weight of recent and past history in the prediction. If  $\alpha$ =0, then  $T_{n+1}$ = $t_n$ , and only the most recent CPU burst matters.

More commonly,  $\alpha$ =1/2, so recent history and past history are equally weighted. The initial  $T_0$  can be defined as a constant or as an overall system average. Figure 6.1 shows an exponential average with  $\alpha$ =1/2 and  $T_0$ =10.To understand the behavior of the exponential average, the formula for  $T_{n+1}$  can be expanded by substituting for  $T_n$ , to find

$$T_{n+1} = \alpha t_n + (1 - \alpha) \alpha t_{n-1} + ... + (1 - \alpha)^j \alpha t_{n-j} + ... + (1 - \alpha)^{n+1} T_0$$

Since both  $\alpha$  and  $(1-\alpha)$  are less than or equal to 1, each successive term has less weight than its predecessor.

The SJF algorithm may be either preemptive or non-preemptive. The choice arises when a new process arrives at the ready queue while a previous process is executing. The new process may have a shorter next CPU burst than what is left of the currently executing process. A preemptive SJF algorithm will preempt the currently executing process, whereas a non-preemptive SJF algorithm will allow the currently running process to finish its CPU burst. Preemptive SJF scheduling is sometimes called shortest-remaining-time-first scheduling.

Consider the following four processes, with length of the CPU burst time given in milliseconds:

| <u>Process</u> | <u>Arrival Time</u> | <u>Burst Time</u> |
|----------------|---------------------|-------------------|
| $\mathbf{P}_1$ | 0                   | 8                 |
| $P_2$          | 1                   | 4                 |
| $P_3$          | 2                   | 9                 |
| $P_4$          | 3                   | 5                 |

If the processes arrive at the ready queue at the times shown and need the indicated burst times, then the resulting preemptive SJF schedule is as depicted in the following Gantt chart:



Process  $P_1$  is started at time 0, since it is the only process in the queue. Process  $P_2$  arrives at time 1. The remaining time for process  $P_1$  (7 milliseconds) is larger than the time required by process  $P_2$  (4 milliseconds), so process  $P_1$  is preempted, and process  $P_2$  is scheduled. the average waiting time for this example is ((10-1)+(1-1)+(17-2)+(5-3))/4=26/4=6.5 milliseconds. A non-preemptive SJF scheduling would result in an average waiting time of 7.75 milliseconds.

# **Priority Scheduling**

The SJF algorithm is a special case of the general priority-scheduling algorithm. A priority is associated with each process, and the CPU is allocated to the process with the highest priority. Equal-priority processes are scheduled in FCFS order.

An SJF algorithm is simply a priority algorithm where the priority (p) is the inverse of the (predicted) next CPU burst. The larger the CPU burst, the lower the priority, and vice versa.

Priorities are generally some fixed range of numbers, such as 0 to 7, or 0 to 4, 095. However, there is no general agreement on whether 0 is the highest or lowest priority. Some systems use low numbers to represent low priority; others use low numbers for high priority. This difference can lead to confusion. In this description, low priority numbers are used to represent high priority.

Consider the following set of processes, assumed to have arrived at time 0, in the order  $P_1, P_2, ..., P_5$ , with the length of the CPU burst time given in milliseconds as an example:

| <u>Process</u>   | Burst Time | <b>Priority</b> |
|------------------|------------|-----------------|
| $\mathbf{P}_{1}$ | 10         | 3               |
| $P_2$            | 1          | 1               |
| $P_3$            | 2          | 4               |
| $P_4$            | 1          | 5               |
| $P_5$            | 5          | 2               |

Using priority scheduling, these processes are scheduled according to the following Gantt chart:

| $P_2$ | P <sub>5</sub> | P <sub>1</sub> | P <sub>3</sub> | P <sub>4</sub> |    |
|-------|----------------|----------------|----------------|----------------|----|
| 0     | 1 (            |                | 16             | 18             | 19 |

The average waiting time is 8.2 milliseconds.

Priorities can be defined either internally or externally. Internally defined priorities use some measurable quantity or quantities to compute the priority of a process. For example, time limits, memory requirements, the number of open files, and the ratio of average I/O burst to average CPU burst have been

used in computing priorities. External priorities are set by criteria that are external to the operating system, such as the importance of the process, the type and amount of funds being paid for computer use, the department sponsoring the work, and other, often political factors.

Priority scheduling can be either preemptive or non-preemptive. When a process arrives at the ready queue, its priority is compared with priority of the currently running process. A preemptive priority scheduling algorithm will preempt the CPU if the priority of the newly arrived process is higher than the priority of the currently running process. A non-preemptive priority scheduling algorithm will simply put the new process at the head of the ready queue.

# **Round Robin Scheduling**

The round robin (RR) scheduling algorithm is designed especially for time-sharing systems. It is similar to FCFS scheduling, but preemption is added to switch between processes. A small unit of time called a time quantum (or time slice) is defined. A time quantum is generally from 10 to 100 milliseconds. The ready queue is treated as a circular queue. The CPU scheduler goes around the ready queue, allocating the CPU to each process for a time interval of up to 1 time quantum.

To implement RR scheduling, the ready queue is kept as a FIFO queue of processes. New processes are added to the tail of the ready queue. The CPU scheduler picks the first process from the ready queue, sets a timer to interrupt after 1 time quantum, and dispatches the process.

One of two things will then happen. The process may have a CPU burst of less than 1 time quantum. In this case, the process itself will release the CPU voluntarily. The scheduler will then proceed to the next process in thread queue. Otherwise, if the CPU burst of the currently running process is longer than 1 time quantum, the timer will go off and will cause an interrupt to the operating system. A context switch will be executed, and the process will be put at the tail of the ready queue. The CPU scheduler will then select the next process in the ready queue.

Consider the following set of processes that arrive at time 0, with the length of the CPU burst time given in milliseconds:

| <u>Process</u> | <b>Burst Time</b> |
|----------------|-------------------|
| $\mathbf{P}_1$ | 24                |
| $P_2$          | 3                 |
| $P_3$          | 3                 |

If a time quantum of 4 milliseconds is used, then process  $P_1$  gets the first 4 milliseconds. Since it requires another 20 milliseconds, it is preempted after the first time quantum, and the CPU is given to the next process in the queue, process  $P_2$ . Since process  $P_2$  does not need 4 milliseconds, it quits before its time

quantum expires. The CPU is then given to the next process, process  $P_3$ . Once each process has received 1 time quantum, the CPU is returned to process  $P_1$  for an additional time quantum. The resulting RR schedule is



The average waiting time is 17/3=5.66 milliseconds.

In RR scheduling algorithm, no process is allocated the CPU for more than 1 time quantum in a row. If a process's CPU burst exceeds 1 time quantum, that process is preempted and is put back in the ready queue. The RR scheduling algorithm is preemptive.

If there are n processes in the ready queue and the time quantum is q, then each process gets 1/n of the CPU time in chunks of at most q time units. Each process must wait no longer than (n-1)×q time units until its next time quantum. For example, then each process will get up to 20 milliseconds every 100 milliseconds.

**Conclusion:** Thus, we have studied various scheduling algorithms.

# **QUESTIONS:**

- 1. Scheduling? List types of scheduler & scheduling.
- 2. List and define scheduling criteria.
- 3. Define preemption & non-preemption.
- 4. State FCFS, SJF, Priority & Round Robin scheduling.
- 5. Compare FCFS, SJF, RR, Priority

# **ASSIGNMENT NO:08 – Group B**

**Assignment Title:** Implement a program to simulate Memory placement strategies – best fit, first fit, next fit and worst fit.

**Problem Statement:** Write a program to simulate Memory placement strategies – best fit, first fit, next fit and worst fit.

# **Objectives:**

- 1. To understand concept of memory placement technique.
- 2. To implement concept of memory placement technique.

# Theory:

- First Fit:-
  - Advantages
  - It is the fastest search as it searches only the first block i.e. enough to assign a process.
  - It may have problems of not allowing processes to take space even if it was possible to allocate. Consider the above example; process number 4 (of size 426) does not get memory. However it was possible to allocate memory if we had allocated using best fit allocation [block number 4 (of size 300) to process 1, block number 2 to process 2, block number 3 to process 3 and block number 5 to process 4].
  - Implementation:
  - Input memory blocks with size and processes with size.
  - Initialize all memory blocks as free.
  - Start by picking each process and check if it can, be assigned to current block.
  - If size-of-process <= size-of-block if yes then assign and check for next process.
  - If not then keep checking the further blocks.



# **Next Fit**

Next fit is a modified version of 'first fit'. It begins as the first fit to find a free partition but when called next time it starts searching from where it left off, not from the beginning. This policy makes use of a roving pointer. The pointer moves along the memory chain to search for a next fit. This helps in, to avoid the usage of memory always from the head (beginning) of the free block chain.

# • Advantages:-

- First fit is a straight and fast algorithm, but tends to cut large portion of free parts into small pieces due to which, processes that need a large portion of memory block would not get anything even if the sum of all small pieces is greater than it required which is so-called external fragmentation problem.
- Another problem of the first fit is that it tends to allocate memory parts at the beginning of the
  memory, which may lead to more internal fragments at the beginning. Next fit tries to address this
  problem by starting the search for the free portion of parts not from the start of the memory, but from
  where it ends last time.
- Next fit is a very fast searching algorithm and is also comparatively faster than First Fit and Best Fit Memory Management Algorithms.

# • Implementation

Input the number of memory blocks and their sizes and initializes all the blocks as free.

- Input the number of processes and their sizes.
- Start by picking each process and check if it can be assigned to the current block, if yes, allocate it the required memory and check for next process but from the block where we left not from starting.
- If the current block size is smaller then keep checking the further blocks.



# Worst Fit

Allocates a process to the partition which is largest sufficient among the freely available partitions available in the main memory. If a large process comes at a later stage, then memory will not have space to accommodate it.

# Implementation:

- Input memory blocks and processes with sizes.
- Initialize all memory blocks as free.
- Start by picking each process and find the maximum block size that can be assigned to current process i.e., find max(bockSize[1], blockSize[2], .....blockSize[n]) > processSize[current], if found then assign it to the current process.
- If not then leave that process and keep checking the further processes.



**Conclusion:** Processes and files allocated to free blocks. List of processes and files which are not allocated memory. The remaining free space list left out after performing allocation..

# **QUESTIONS:**

1.

# **ASSIGNMENT NO: 09 – Group B**

Assignment Title: Implementation of concept called Paging, using simulation.

**Problem Statement:** Write a Java Program (using OOP features) to implement paging simulation using

- 1. Least Recently Used (LRU)
- 2. Optimal algorithm

# 1.2 Objectives:

- 1. To understand concept of paging.
- 2. To learn and paging techniques.
- 1.3 Theory: Paging: Paging is a memory management scheme that eliminates the need for contiguous allocation of physical memory. This scheme permits the physical address space of a process to be non – contiguous.
  - Logical Address or Virtual Address (represented in bits): An address generated by the CPU
  - Logical Address Space or Virtual Address Space( represented in words or bytes): The set of all logical addresses generated by a program
  - Physical Address (represented in bits): An address actually available on memory unit
  - Physical Address Space (represented in words or bytes): The set of all physical addresses corresponding to the logical addresses

# **Example:**

- If Logical Address = 31 bit, then Logical Address Space = 2<sup>31</sup> words = 2 G words (1 G = 2<sup>30</sup>)
   If Logical Address Space = 128 M words = 2<sup>7</sup> \* 2<sup>20</sup> words, then Logical Address = log<sub>2</sub> 2<sup>27</sup> = 27 bits
- If Physical Address = 22 bit, then Physical Address Space =  $2^{22}$  words = 4 M words (1 M =  $2^{20}$ )
- If Physical Address Space = 16 M words =  $2^4 * 2^{20}$  words, then Physical Address =  $\log_2 2^{24} = 24$  bits

The mapping from virtual to physical address is done by the memory management unit (MMU) which is a hardware device and this mapping is known as paging technique.

- The Physical Address Space is conceptually divided into a number of fixed-size blocks, called
- The Logical address Space is also splitted into fixed-size blocks, called **pages**.
- Page Size = Frame Size

Let us consider an example:

- Physical Address = 12 bits, then Physical Address Space = 4 K words
- Logical Address = 13 bits, then Logical Address Space = 8 K words
- Page size = frame size = 1 K words (assumption)



Address generated by CPU is divided into

- Page number(p): Number of bits required to represent the pages in Logical Address Space or Page number
- Page offset(d): Number of bits required to represent particular word in a page or page size of Logical Address Space or word number of a page or page offset.

Physical Address is divided into

- **Frame number(f):** Number of bits required to represent the frame of Physical Address Space or Frame number.
- Frame offset(d): Number of bits required to represent particular word in a frame or frame size of Physical Address Space or word number of a frame or frame offset.

The hardware implementation of page table can be done by using dedicated registers. But the usage of register for the page table is satisfactory only if page table is small. If page table contain large number of entries then we can use TLB(translation Look-aside buffer), a special, small, fast look up hardware cache.

- The TLB is associative, high speed memory.
- Each entry in TLB consists of two parts: a tag and a value.
- When this memory is used, then an item is compared with all tags simultaneously. If the item is found, then corresponding value is returned.

Main memory access time = m

If page table are kept in main memory,

Effective access time = m(for page table) + m(for particular page in page table)





# **Page Replacement Algorithms:**

In a operating systems that use paging for memory management, page replacement algorithm are needed to decide which page needed to be replaced when new page comes in. Whenever a new page is referred and not present in memory, page fault occurs and Operating System replaces one of the existing pages with newly needed page. Different page replacement algorithms suggest different ways to decide which page to replace. The target for all algorithms is to reduce number of page faults.

**Page Fault** – A page fault is a type of interrupt, raised by the hardware when a running program accesses a memory page that is mapped into the virtual address space, but not loaded in physical memory.

# **Page Replacement Algorithms:**

1. **Least Recently Used:**—In this algorithm page will be replaced which is least recently used.

# Algorithm:

Let **capacity** be the number of pages that memory can hold. Let **set** be the current set of pages in memory.

- 1- Start traversing the pages.
  - i) If set holds less pages than capacity.
    - a) Insert page into the set one by one until the size of set reaches capacity or all page requests are processed.
    - b) Simultaneously maintain the recent occurred index of each page in a map called **indexes**.
    - c) Increment page fault
  - ii) Else

If current page is present in **set**, do nothing.

# Else

- a) Find the page in the set that was least recently used. We find it using index array.
- We basically need to replace the page withminimum index.
- b) Replace the found page with current page.
- c) Increment page faults.
- d) Update index of current page.
- 2. Return page faults.

Let say the page reference string 7 0 1 2 0 3 0 4 2 3 0 3 2. Initially we have 4 page slots empty. Initially all slots are empty, so when 7 0 1 2 are allocated to the empty slots —> 4 Page faults 0 is already their so —>0 Page fault.

When 3 came it will take the place of 7 because it is least recently used —>1 Page fault 0 is already in memory so —> 0 Page fault.

4 will takes place of 1 —> 1 Page Fault

Now for the further page reference string —> 0 Page fault because they are already available in the memory.

| г 1         | T 42 1      | C            |                | 1      | 1    |       | 1 ' | ı     | 1                 | 1   | 1      | 1 /1  | •    | C 41    | C 1 /      | 4   |
|-------------|-------------|--------------|----------------|--------|------|-------|-----|-------|-------------------|-----|--------|-------|------|---------|------------|-----|
| Example-    | Let's have  | a reterence  | e string: a    | n.c    | . a. | c.a.  | a.  | n. e. | n.a.              | n.c | . a an | a the | Size | or the  | trame be 4 | 4.  |
| - Linuingio | , Loub mare | a rererence. | D D CI III CC. | , ,, , | , ~, | ·, ~, | ••• | ·, ·, | $\sim$ , $\sim$ , | ~ · | ,      |       | DIL  | 01 1110 | manie ce   | • • |

| Tim | ne req. | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 |
|-----|---------|---|---|---|---|---|---|---|---|---|----|----|----|----|----|
| ١,  | / _     |   |   |   |   |   |   |   |   |   |    |    |    |    |    |
|     | Page    |   |   |   |   |   |   |   |   |   |    |    |    |    |    |
| fra | mes     | a | ь | С | d | С | a | d | b | e | ь  | a  | b  | С  | d  |
|     |         |   |   |   |   |   |   |   |   |   |    |    |    |    |    |
| 0   | a       | a | a | a | a | a | a | a | a | a | e  | e  | e  | e  | e  |
| 1   | b       |   | b | b | b | b | b | b | b | b | b  | a  | a  | a  | a  |
| 2   | С       |   |   | c | С | c | С | c | c | e | c  | c  | Ъ  | ь  | d  |
| 3   | d       |   |   |   | d | d | d | d | d | d | d  | d  | d  | c  | c  |
| FA  | ULTS    | x | x | x | x |   |   |   |   | x |    |    |    | x  | x  |

There are 7 page faults using LRU algorithm.

2. <u>Optimal Page replacement</u>: – In this algorithm, pages are replaced which are not used for the longest duration of time in the future. Let us consider page reference string 7 0 1 2 0 3 0 4 2 3 0 3 2 and 4 page slots.

Initially all slots are empty, so when 7 0 1 2 are allocated to the empty slots —> 4 Page faults 0 is already there so —>0 Page fault.

When 3 came it will take the place of 7 because it's not used for longest duration in future. ->1 Page fault. 0 is already there so —> 0 Page fault.

4 will take place of 1 —> 1 Page Fault.

Now for the further page reference string —> 0 Page fault because they are already available in the memory.

Example-2, Let's have a reference string: a, b, c, d, c, a, d, b, e, b, a, b, c, d and the size of the frame be 4.

| Tim | ne req.     | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 |
|-----|-------------|---|---|---|---|---|---|---|---|---|----|----|----|----|----|
|     | Page<br>mes | a | ь | c | d | с | a | d | b | e | ь  | a  | ь  | с  | d  |
| 0   | a           | a | a | a | a | a | a | a | a | a | e  | e  | e  | e  | d  |
| 1   | b           |   | b | b | b | b | b | b | b | b | b  | a  | a  | a  | a  |
| 2   | С           |   |   | c | c | c | c | c | c | c | С  | c  | b  | b  | b  |
| 3   | d           |   |   |   | d | d | d | d | d | e | d  | d  | d  | С  | c  |
| FA  | ULTS        | x | x | x | X |   |   |   |   | x |    |    |    |    | x  |

There are 6 page faults using optimal algorithm.

Optimal page replacement is perfect, but not possible in practice as operating system cannot know future requests. The use of Optimal Page replacement is to set up a benchmark so that other replacement algorithms can be analyzed against it.

**Conclusion:** Thus, we have studied various paging techniques.

# **ASSIGNMENT NO: 10 – Group B**

Assignment Title: Perform Implementation of Deadlock avoidance algorithm, i.e Bankers Algorithms.

**Problem Statement:** Write a Java program to implement Banker's Algorithm.

# **Objectives:**

- 1. To understand concept of resource allocation and deadlock avoidance.
- 2. To learn and use Bankers algorithms.

# Theory: Bankers Algorithm:-

The resource-allocation graph algorithm is not applicable to a resource-allocation system with multiple instances of each resource type. The deadlock-avoidance algorithm that we describe next is applicable to such a system, but is less efficient than the resource-allocation graph scheme. This algorithm is commonly known as the banker's algorithm. The name was chosen because this algorithm could be used in a banking system to ensure that the bank never allocates its available cash such that it can no longer satisfy the needs of all its customers.

When a new process enters the system, it must declare the maximum number of instances of each resource type that it may need. This number may not exceed the total number of resources in the system. When a user requests a set of resources, the system must determine whether the allocation of these resources will leave the system in a safe state. If it will, the resources are allocated; otherwise, the process must wait until some other process releases enough resources.

# Algorithm

Several data structures must be maintained to implement the banker's algorithm. These data structures encode the state of the resource-allocation system. Let n be the number of processes in the system and m be the number of resource types. We need the following data structures:

- 1. **Available:** A vector of length m indicates the number of available resources of each type. If Available[j]=k, there are k instances of resource type  $R_j$  available.
- 2. **Max:** An  $n \times m$  matrix defines the maximum demand of each process. If Max[i,j]=k, then process  $P_i$  may request at most k instances of resource type  $R_j$ .
- 3. **Allocation:** An  $n \times m$  matrix defines the number of resources of each type currently allocated to each process. If *Allocation* [i,j]=k, then process  $P_i$  is currently allocated k instances of resource type  $R_j$ .
- 4. Need: An  $n \times m$  matrix indicates the remaining resource need of each process. If Need [i,j]=k, then process  $P_i$  may need k more instances of resource type  $R_j$  to complete its task.

Need[i,j]=Max[i,j]-Allocation[i,j]

These data structures vary over time in both size and value.

To simplify the presentation of the banker's algorithm, let us establish some notation. Let X and Y be vectors of length n,  $X \le Y$  if and only if  $X[i] \le Y[i]$  for all i=1,2,...,n. For example, if X=(1,7,3,2) and Y=(0,3,2,1), then  $Y \le X$ . Y < X if  $Y \le X$  and  $Y \ne X$ .

We can treat each row in the matrices Allocation and Need as vectors and refer to them as  $Allocation_i$  and  $Need_i$ , respectively. The vector  $Allocation_i$  specifies the resources currently allocated to process  $P_i$ ; the vector  $Need_i$  specifies the additional resources that process  $P_i$ may still request to complete its task.

# Safety Algorithm

The algorithm for finding out whether or not a system is in a safe state can be:

Let Work and Finish be vectors of length m and n, respectively. Initialize Work:=Available and Finish [i]:=false for i=1,2,...,n

- 1. Find an i such that both
  - a. Finish[i]=false
  - b. Needi<Work

If no such i exists, go to step 4.

2.  $Work:=Work+Allocation_i$ 

Finish[i]:=true go to step 2.

3. If Finish[i] = true for all i, then the system is in a safe state.

This algorithm may require an order of  $m \times n^2$  operations to decide whether a state is safe.

# **Resource-Request Algorithm**

Let  $Request_i$  be the request vector for process  $P_i$ . If  $Request_i[j]=k$ , then process  $P_i$  wants k instances of resource type  $R_i$ . When a request for resources is made by process  $P_i$ , the following actions are taken:

- 1. If *Requesti≤Needi*, go to step2. Otherwise, raise an error condition, since the process has exceeded its maximum claim.
- 2. If *Requesti≤Available*, go to step3. Otherwise, *Pi* must wait, since the resources are not available.
- 3. Have the system pretend to have allocated the requested resources to process *Pi* by modifying the state as follows:

Available:=Available-Request<sub>i</sub>; Allocation<sub>i</sub>:=Allocation<sub>i</sub>+Request<sub>i</sub>; Need<sub>i</sub>:=Need<sub>i</sub>-Request<sub>i</sub>;

If the resulting resource-allocation state is safe, the transaction is completed and process  $P_i$  is allocated its resources. If the new state is unsafe, then  $P_i$  must wait for  $Request_i$  and the old resources-allocation state is restored.

**Example:** The system with five processes  $P_0$  through  $P_4$  and three resource types A,B,C. Resource type A has 10 instances, resource type B has 5 instances, and

resource type C has 7 instances. Suppose that, at time  $T_0$ , the following snapshot of the system has been taken:

|       | <u>Allocation</u> | <u> Max</u> | <u>Available</u> |
|-------|-------------------|-------------|------------------|
|       | A B C             | ABC         | A B C            |
| $P_0$ | 0 1 0             | 7 5 3       | 3 3 2            |
| $P_1$ | 2 0 0             | 3 2 2       |                  |
| $P_2$ | 3 0 2             | 9 0 2       |                  |
| $P_3$ | 2 1 1             | 2 2 2       |                  |
| $P_4$ | 0 0 2             | 4 3 3       |                  |

The content of the matrix *Need* is defined to be *Max-Allocation* and is

<u>Need</u>

 We claim that the system is currently in a safe state. The sequence

<P<sub>1</sub>, P<sub>3</sub>, P<sub>4</sub>,P<sub>2</sub>, P<sub>0</sub>> satisfies the safety criteria. Suppose now that process P<sub>1</sub> requests one additional instance of resource type A and two instances of resource type C, so Request<sub>1</sub>=(1,0,2). To decide whether this request can be immediately granted, we first check that Request<sub>1</sub>  $\le$ Available which is true. Then this request has been fulfilled, and the following new state is arrived:

|       | <u>Allocation</u> | <u>Need</u> | <u>Available</u> |  |  |
|-------|-------------------|-------------|------------------|--|--|
|       | A B C             | A B C       | A B C            |  |  |
| $P_0$ | 0 1 0             | 7 4 3       | 2 3 0            |  |  |
| $P_1$ | 3 0 2             | 0 2 0       |                  |  |  |
| $P_2$ | 3 0 2             | 6 0 0       |                  |  |  |
| $P_3$ | 2 1 1             | 0 1 1       |                  |  |  |
| $P_4$ | 0 0 2             | 4 3 1       |                  |  |  |

We must determine whether this new system state is safe. To do so, safety algorithm should be executed and the sequence is found to be  $\langle P_1, P_3, P_4, P_0, P_2 \rangle$  satisfies our safety requirement. Hence, the request of process P1 can be granted immediately.

**Conclusion:** Thus, we have studied deadlock avoidance using Bankers algorithms.

# **QUESTIONS:**

- 1. Which one of the following is the deadlock avoidance *algorithm*?
- a) Banker's algorithm b) round-robin algorithm c) elevator algorithm d) karn's algorithm.
- 2. What is deadlock? Why it occurs in system?
- 3. What is Deadlock prevention
- 4. What is Deadlock Avoidance
- 5. Explain Bankers algorithm with one example.